



XX XX IIIII DDDDDDDDD RRRRRRRRR VV VV EEEEEEEEEE RRRRRRRRR  
XX XX IIIII DDDDDDDDD RRRRRRRRR VV VV EEEEEE RRRRRRRRR  
XX XX IIIII DD RR RR VV VV EEEEEE RRRRRRRRR  
XX XX IIIII DD RR RR VV VV EEEEEE RRRRRRRRR  
XX XX IIIII DD RR RR VV VV EEEEEE RRRRRRRRR  
XX XX IIIII DD RR RR VV VV EEEEEE RRRRRRRRR  
XX XX IIIII DD RR RR VV VV EEEEEE RRRRRRRRR  
XX XX IIIII DD RR RR VV VV EEEEEE RRRRRRRRR  
XX XX IIIII DD RR RR VV VV EEEEEE RRRRRRRRR  
XX XX IIIII DD RR RR VV VV EEEEEE RRRRRRRRR  
XX XX IIIII DDDDDDDDD RR RR VV VV EEEEEE RRRRRRRRR  
XX XX IIIII DDDDDDDDD RR RR VV VV EEEEEE RRRRRRRRR

MM MM AAAAAA RRRRRRRR  
MM MM AAAAAA RRRRRRRR  
MM MM AA AA RR RR  
MM MM AA AA RR RR  
MM MM AA AA RR RR  
MM MM AA AA RRRRRRRR  
MM MM AA AA RRRRRRRR  
MM MM AAAAAAAA RR RR  
MM MM AAAAAAAA RR RR  
MM MM AA AA RR RR

:TITLE XIDRIVER - VAX/VMS DMF32 PARALLEL PORT DRIVER  
:IDENT 'V04-001'

\*\*\*\*\*  
\* COPYRIGHT (c) 1978, 1980, 1982, 1984 BY  
\* DIGITAL EQUIPMENT CORPORATION, MAYNARD, MASSACHUSETTS.  
\* ALL RIGHTS RESERVED.

\* THIS SOFTWARE IS FURNISHED UNDER A LICENSE AND MAY BE USED AND COPIED  
\* ONLY IN ACCORDANCE WITH THE TERMS OF SUCH LICENSE AND WITH THE  
\* INCLUSION OF THE ABOVE COPYRIGHT NOTICE. THIS SOFTWARE OR ANY OTHER  
\* COPIES THEREOF MAY NOT BE PROVIDED OR OTHERWISE MADE AVAILABLE TO ANY  
\* OTHER PERSON. NO TITLE TO AND OWNERSHIP OF THE SOFTWARE IS HEREBY  
\* TRANSFERRED.

\* THE INFORMATION IN THIS SOFTWARE IS SUBJECT TO CHANGE WITHOUT NOTICE  
\* AND SHOULD NOT BE CONSTRUED AS A COMMITMENT BY DIGITAL EQUIPMENT  
\* CORPORATION.

\* DIGITAL ASSUMES NO RESPONSIBILITY FOR THE USE OR RELIABILITY OF ITS  
\* SOFTWARE ON EQUIPMENT WHICH IS NOT SUPPLIED BY DIGITAL.

\*\*\*\*\*

++  
FACILITY:

VAX/VMS Executive, I/O Drivers

ABSTRACT:

This driver is an example driver for the DMF32 parallel port.  
This driver implements the DR11C compatibility mode on the device.  
It does not implement the silo or DMA options, but serves as a  
template to which such features could be added.

This module contains the DMF32 PARALLEL PORT driver:

Tables for loading and dispatching  
Controller initialization routine  
FDT routine  
The start I/O routine  
The interrupt service routine  
Device specific Cancel I/O

ENVIRONMENT:

Kernal Mode, Non-paged

AUTHOR:

Jake VanNoy January 1982

## MODIFIED BY:

|         |         |                                                           |             |
|---------|---------|-----------------------------------------------------------|-------------|
| V04-001 | JLV0396 | Jake VanNoy                                               | 6-SEP-1984  |
|         |         | Add AVL to DEVCHAR.                                       |             |
| V03-005 | JLV0385 | Jake VanNoy                                               | 23-JUL-1984 |
|         |         | Add DPTSM_SVP to DPT.                                     |             |
| V03-004 | JLV0341 | Jake VanNoy                                               | 28-MAR-1984 |
|         |         | Correct Device IPL.                                       |             |
| V03-003 | WMM0002 | Bill Matthews                                             | 16-Feb-1984 |
|         |         | Second part of change for edit WMM0001.                   |             |
| V03-002 | WMM0001 | Bill Matthews                                             | 19-Dec-1983 |
|         |         | Added code to support new IDB fields IDB\$B_COMBO_VECTOR  |             |
|         |         | and IDB\$B_COMBO_CSR_OFFSET for determining the main CSR  |             |
|         |         | address and loading the soft vector for the combo device. |             |
| V03-001 | KDM0002 | Kathleen D. Morse                                         | 28-Jun-1982 |
|         |         | Added \$DCDEF and \$DYNDEF.                               |             |

--

++ .SBTTL Description of Interface

The DMF32 Parallel Port interface is a 16 bit parallel port for interfacing to a user device. It includes a DR11C compatibility mode (used for word mode within this driver), a silo (buffered) mode (not implemented by this driver), and a DMA mode (also not implemented by this driver). The interface looks like the following:



++

This driver may be tested using the following configuration of two DMF32's:  
The control lines (CTRL 0 and 1) should be tied into request lines (REQ A  
and B) on the other device. Setting CTRL 0 on the first device causes an  
interrupt on REQ A on the second device (provided interrupt enable A is set).



--

## SBTTL Documentation on interface

++

The DMF32 parallel port exchanges one 16-bit word at a time. A single QIO request transfers a buffer of data, with an interrupt requested for each word.

For each buffer of data transferred, the DMF32 parallel port allows for the exchange of additional bits of information: the Control and Status Register (CSR) function (CTRL) and status (REQUEST) bits. These bits are accessible to an application process through the device driver QIO interface. The CTRL bits are labeled CTRL 0 and CTRL 1. The REQUEST bits are labeled REQUEST A and REQUEST B.

The user device interfaced to the DMF32 parallel port interprets the value of the two CTRL bits. The QIO request that initiates the transfer specifies the IOSM\_SETFNCT modifier to indicate a change in the value for the CTRL bits. The P4 argument of the request specifies this value. P4 bits 0 and 1 correspond to CTRL bits 0 and 1 respectively. Bits 2 through 31 are not used. If required, the CTRL bits must be set for each request. The CTRL bits set in the CSR are passed directly to the user device.

The device class for the DMF32 parallel port is DCS\_REALTIME and the device type is DTS\_XI\_DR11C. The DMF32 parallel port driver does not use the default buffer size field. The value of this field is set to 65,535. This driver defines no device-dependent characteristics.

The DMF32 parallel port can perform logical, virtual, and physical I/O operations. The basic I/O functions are read, write, set mode, and set characteristics.

| Function Code and Arguments   | Function Modifiers                       | Function                                           |
|-------------------------------|------------------------------------------|----------------------------------------------------|
| IOS_READBLK P1,P2,-<br>P3,P4  | IOSM_SETFNCT<br>IOSM_RESET<br>IOSM_TIMED | Read block !                                       |
| IOS_WRITEBLK P1,P2,-<br>P3,P4 | IOSM_SETFNCT<br>IOSM_RESET<br>IOSM_TIMED | Write logical block                                |
| IOS_SETMODE P1,P3             | IOSM_ATTNAST                             | Set PORT characteristics for subsequent operations |
| IOS_SETCHAR P1,P3             | IOSM_ATTNAST                             | Set PORT characteristics for subsequent operations |

Not in above table are functions IOS\_READPBLK, IOS\_READVBLK, WRITEPBLK

: and WRITELBLK. There is no functional difference in these operations.  
Although the DMF32 parallel port does not differentiate between logical,  
virtual, and physical I/O functions (all are treated identically), the  
user must have the required privilege to issue a request.

The function-dependent arguments for the read and write function codes are:

- o P1 -- the starting virtual address of the buffer that is to receive data in the case of a read operation; or, in the case of a write operation, the virtual address of the buffer that is to send data to the DMF32 parallel port. Modify access to the buffer, rather than read or write access, is checked for all block mode read and write requests.
- o P2 -- the size of the data buffer in bytes, that is, the transfer count. Since the DMF32 parallel port performs word transfers, the transfer count must be an even value.
- o P3 -- the timeout period for this request (in seconds). The value specified must be equal to or greater than 2. IOSM\_TIMED must be specified. The default timeout value for each request is 10 seconds.
- o P4 -- the value of the DMF32 parallel port Command and Status Register (CSR) function (CTRL) bits to be set. If IOSM\_SETFNCT is specified, the low-order three bits of P4 (2:0) are written to CSR CTRL bits 1:0 (respectively) at the time of transfer.

The transfer count specified by the P2 argument must be an even number of bytes. If an odd number or more than 65534 bytes is specified, an error (SSS\_BADPARAM) is returned in the I/O status block (IOSB). If the transfer count is 0, the driver will transfer no data. However, if IOSM\_SETFNCT is specified and P2 is 0, the driver will set the CTRL bits in the DMF32 parallel port CSR, and return the current CSR status bit values in the IOSB.

The read and write QIO functions can take three function modifiers:

- o IOSM\_SETFNCT - set the function (CTRL) bits in the DMF32 parallel port CSR before the data transfer is initiated. The low-order two bits of the P4 argument specify the CTRL bits. The user device that interfaces the DMF32 PARALLEL PORT receives the CTRL bits directly and their value is interpreted entirely by the device.

If an unsolicited interrupt is received from the DMF32 parallel port, no read or write request is posted, and the next request is for a word mode read, the driver will return the word read from the DMF32 parallel port INBUF and store it in the first word of the user's buffer. In this case the driver does not wait for an interrupt.

- o IOSM\_TIMED - set the device timeout interval for the data transfer request. The P3 argument specifies the timeout interval value in seconds. For consistent results, this

value must be equal to or greater than 2.

- o **IOSM\_RESET** - perform a device reset to the DMF32 parallel port before any I/O operation is initiated. This function does not affect any other device on the system or on the DMF32.

The set mode and characteristic function codes are:

- o **IOS\_SETMODE**
- o **IOS\_SETCHAR**

These functions take the following device/function-dependent arguments:

- o **P1** - the virtual address of a quadword characteristics buffer. If the function modifier **IOSM\_ATTNAST** is specified, **P1** is the address the AST service routine. In this case, if **P1** is 0, all attention ASTs are disabled.
- o **P3** - the access mode to deliver the AST (maximized with the requestor's access mode). If **IOSM\_ATTNAST** is not specified, **P3** is ignored.

Figure 3-4 shows the quadword **P1** characteristics buffer for **IOS\_SETMODE** and **IOS\_SETCHAR**.



The **IOS\_SETMODE** and **IOS\_SETCHAR** function codes can take the following function modifier:

- o **IOSM\_ATTNAST** - enable attention AST

This function modifier allows the user process to queue an attention AST for delivery when an asynchronous or unsolicited condition is detected by the DMF32 parallel port driver. Unlike ASTs for other I/O functions, use of this function modifier does not increment the I/O count for the requesting process or lock pages in memory for I/O buffers. There must be an AST quota for each AST.

Attention ASTs are delivered under the following conditions:

- o An unsolicited interrupt from the DMF32 parallel port occurs.
- o An attention AST is queued and a previous unsolicited interrupt has not been acknowledged.

: The \$CANCEL system service is used to flush attention ASTs for a specific channel.

: IOS\_SETMODE!IOSM\_ATTNAST and IOS\_SETCHAR!IOSM\_ATTNAST are one-time AST enables; they must be explicitly re-enabled once the AST has been delivered if the user desires notification of the next interrupt. Use of this function modifier does not update the device characteristics.

: After the AST is delivered, the QIO astprm parameter contains the contents of the DMF32 parallel port CSR in the low two bytes and the value read from the DMF32 parallel port INBUF in the high two bytes.

: On completion of each read or write request, the I/O status block is filled with system and DMF32 parallel port status information.



## .SBTTL External and local symbol definitions

## ; External symbols

|           |                                 |
|-----------|---------------------------------|
| \$ACBDEF  | : AST control block             |
| \$CRBDEF  | : Channel request block         |
| \$DCDEF   | : Device types                  |
| \$DDDBDEF | : Device data block             |
| \$DPTDEF  | : Driver prolog table           |
| \$DYNDEF  | : Dynamic data structure types  |
| \$IDBDEF  | : Interrupt data block          |
| \$IODEF   | : I/O function codes            |
| \$IPLDEF  | : Hardware IPL definitions      |
| \$IRPDEF  | : I/O request packet            |
| \$PRDEF   | : Internal processor registers  |
| \$PRIDEF  | : Scheduler priority increments |
| \$SSDEF   | : System messages               |
| \$UCBDEF  | : Unit control block            |
| \$VECDEF  | : Interrupt vector block        |

## ; Local symbols

## ; Argument list (AP) offsets for device-dependent QIO parameters

|    |      |                        |
|----|------|------------------------|
| P1 | = 0  | : First QIO parameter  |
| P2 | = 4  | : Second QIO parameter |
| P3 | = 8  | : Third QIO parameter  |
| P4 | = 12 | : Fourth QIO parameter |
| P5 | = 16 | : Fifth QIO parameter  |
| P6 | = 20 | : Sixth QIO parameter  |

## ; Other constants

|                  |         |                                    |
|------------------|---------|------------------------------------|
| XI_DEF_TIMEOUT   | = 10    | : 10 second default device timeout |
| XI_DEF_BUFSIZ    | = 65535 | : Default buffer size              |
| XI_SK_VEC_OFFSET | = 2     | : Vector offset                    |

## ; Macros

; The SETCTRL macro sets the CTRL 0 and 1 lines as they have been  
; specified in P4 in a read or write QIO. They are cleared and a wait  
; occurs before being set. This is because testing for this example  
; driver was done between two DMF32's in word mode, and the delay is so the  
; microcode on the DMF32 can see the control line changes.

.MACRO SETCTRL  
 BICW #XI\_CSR\$M\_CTRL0!XI\_CSR\$M\_CTRL1,XI\_CSR(R4)  
 CRLR -(SP)  
 TIMEWAIT -  
 TIME = #2,-  
 BITVAL = #1,-

SOURCE = (SP),-  
CONTEXT = L,-  
SENSE = .TRUE.  
(SP)+  
TSTL BISW IRP\$L\_SEGVBN(R3),XI\_CSR(R4)  
.ENDM SETCTRL

; UCB\_XI definitions that follow the standard UCB fields

```
$DEFINI UCB
.=UCBSL_DPC+4  :

$DEF  UCBSL_XI_ATTN    .BLKL  1      ; Attention AST queue
$DEF  UCBSL_XI_DPR     .BLKL  1      ; Word count?
$DEF  UCBSW_XI_INBUF   .BLKW  1      ; Input buffer temporary
$DEF  UCBSW_XI_CSR      .BLKW  1      ; CSR temporary
```

; Bit positions for device-dependent status field in UCB (UCBSW\_DEVSTS)

```
$VIELD UCB,0,<-
<ATTNAST,,M>,-
<UNEXPT,,M>-
>
UCBSK_SIZE=.
$DEFEND UCB
```

```
DMF32 Parallel Port CSR definitions
```

```
$DEFINI XI
```

```
$DEF XI_CSR ; Device CSR
```

```
; Bit positions for device control/status register
```

```
$VIELD XI_CSR,0,<- ; Control/status register
  <CTRL0,,M>,- ; Control line 0
  <CTRL1,,M>,- ; Control line 1
  <NPR_P$,,M>,- ; NPR Primary/Secondary
  <INDREG,2,M>,- ; Indirect Register Address
  <INTENB_A,,M>,- ; Interrupt Enable A
  <INTENB_B,,M>,- ; Interrupt Enable B
  <REQ_A,,M>,- ; Request A
  <DONE_P,,M>,- ; Done Primary
  <DONE_S,,M>,- ; Done Secondary
  <,M>,- ; unused
  <FLUSH,,M>,- ; Flush Buffer
  <,M>,- ; unused
  <NXMERR,,M>,- ; Non-existent memory error
  <RESET,,M>,- ; Master Reset
  <REQ_B,,M>,- ; Request B
```

```
>
```

```
XI_CSRSM_IEAB = <XI_CSRSM_INTENB_A>!<XI_CSRSM_INTENB_B> ; Interrupt enable mask
```

```
.BLKW 1 ; Output buffer Register
$DEF XI_OUTBUF .BLKW 1
```

```
; Note that XI_INBUF and XI_MISC are at the same offset
```

```
$DEF XI_INBUF ; Input buffer Register (when read)
$DEF XI_MISC ; Miscellaneous Register (when written)
```

```
; Bit positions for miscellaneous register
```

```
$VIELD XI_MISC,0,<- ; Miscellaneous register
  <MODE,4,M>,- ; Mode
  <,10,M>,- ; unused
  <SECBUF,,M>,- ; Secondary Buffer Address, Bit 17
  <PRIBUF,,M>,- ; Primary Buffer Address, Bit 17
```

```
>
```

```
.BLKW 1 ; Indirect Register
$DEF XI_IND .BLKW 1
```

```
$DEFEND XI ; End of PORT CSR definitions
```

## .SBTTL Device Driver Tables

## ; Driver prologue table

|                |                                   |                                        |
|----------------|-----------------------------------|----------------------------------------|
| DPTAB          | -                                 | : DPT-creation macro                   |
|                | END=XI END,-                      | : End of driver label                  |
|                | ADAPTER=UBA,-                     | : Adapter type                         |
|                | FLAGS=DPT\$M SVP,-                | : Allocate system page table           |
|                | UCBSIZE=UCBSK_SIZE,-              | : UCB size                             |
|                | NAME=XIDRIVER                     | : Driver name                          |
| DPT_STORE INIT |                                   |                                        |
| DPT_STORE      | UCB,UCBSB_FIPL,B,8                | : Start of load initialization table   |
| DPT_STORE      | UCB,UCBSB_DIPL,B,21               | : Device fork IPL                      |
| DPT_STORE      | UCB,UCBSL_DEVCHAR,L,<-            | : Device interrupt IPL                 |
|                | DEVSM_AVL:-                       | : Device characteristics               |
|                | DEVSM_RTM:-                       | : Available                            |
|                | DEVSM_IDV:-                       | : Real Time device                     |
|                | DEVSM_ODV>                        | : input device                         |
| DPT_STORE      | UCB,UCBSB_DEVCLASS,B,DCS          | : output device                        |
| DPT_STORE      | UCB,UCBSB_DEVTYPE,B,DT\$_XI_DR11C | : Device class                         |
| DPT_STORE      | UCB,UCBSW_DEVBUFSIZ,W,-           | : Device Type                          |
|                | XI_DEF_BUFSIZ                     | : Default buffer size                  |
| DPT_STORE      | REINIT                            | : Start of reload initialization table |
| DPT_STORE      | DDB,DDBSL_DDT,D,XISDDT            | : Address of DDT                       |
| DPT_STORE      | CRB,CRBSL_INTD+4,D,-              | : Address of interrupt                 |
|                | XI_INTERRUPT                      | : service routine                      |
| DPT_STORE      | CRB,CRBSL_INTD2+4,D,-             | : Address of interrupt                 |
|                | XI_INTERRUPT                      | : service routine                      |
| DPT_STORE      | CRB,CRBSL_INTD+VECSL_INITIAL,-    | : Address of controller                |
|                | D,XI_CONTROL_INIT                 | : initialization routine               |
| DPT_STORE      | END                               | : End of initialization                |
|                |                                   | : tables                               |

## ; Driver dispatch table

|       |                       |                      |
|-------|-----------------------|----------------------|
| DDTAB | -                     | : DDT-creation macro |
|       | DEVNAM=XI,-           | : Name of device     |
|       | START=XI_START,-      | : Start I/O routine  |
|       | FUNCTB=XI_FUNCTABLE,- | : FDT address        |
|       | CANCEL=XI_CANCEL      | : Cancel I/O routine |

.PAGE

## ; Function dispatch table

XI\_FUNCTABLE: ; FDT for driver

; Valid I/O functions

|         |                                      |  |
|---------|--------------------------------------|--|
| FUNCTAB | -                                    |  |
|         | <READPBLK,READLBLK,READVBLK,-        |  |
|         | WRITEPBLK,WRITELBLK,WRITEVBLK,-      |  |
|         | SETMODE,SETCHAR,SENSEMODE,SENSECHAR> |  |

```
18
FUNCTAB ; No buffered functions
FUNCTAB XI READ WRITE,- ; Device-specific FDT
        <READPBLK,READLBLK,READVBLK, -
         WRITEPBLK,WRITELBLK,WRITEVBLK>
FUNCTAB +EXE$READ,<READPBLK,READLBLK,READVBLK>
FUNCTAB +EXE$WRITE,<WRITEPBLK,WRITELBLK,WRITEVBLK>
FUNCTAB XI SETMODE,<SETMODE,SETCHAR>
FUNCTAB +EXE$SENSEMODE,<SENSEMODE,SENSECHAR>
```

.SBTTL XI\_CONTROL\_INIT, Controller initialization

++ XI\_CONTROL\_INIT, Called when driver is loaded, system is booted, or power failure recovery.

Functional Description:

- 1) Allocates the direct data path permanently
- 2) Assigns the controller data channel permanently
- 3) Clears the Control and Status Register
- 4) If power recovery, requests device time-out

Inputs:

R4 = address of CSR  
R5 = address of IDB  
R6 = address of DDB  
R8 = address of CRB

Outputs:

VECSV\_PATHLOCK bit set in CRBSL\_INTD+VECSB\_DATAPATH  
UCB address placed into IDBSL\_OWNER

--

XI\_CONTROL\_INIT:

MOVL IDBSL\_UCBLST(R5),R0 ; Address of UCB  
MOVL R0, IDBSL\_OWNER(R5) ; Make permanent controller owner  
BISW #UCBSM\_ONLINE, -  
UCBSW\_STS(R0) ; Set device status "on-line"

; If powerfail has occurred and device was active, force device time-out.  
; The user can set his own time-out interval for each request. Time-  
; out is forced so a very long time-out period will be short circuited.

BBS #UCBSV\_POWER, -  
UCBSW\_STS(R0),10\$ ; Branch if powerfail  
BISB #VECSM\_PATHLOCK, -  
CRBSL\_INTD+VECSB\_DATAPATH(R8) ; Permanently allocate direct datapath

10\$:

CVTBL IDBSB\_COMBO\_CSR\_OFFSET(R5),R0 ; GET OFFSET TO MAIN DMF CSR  
SUBB3 IDBSB\_COMBO\_VECTOR\_OFFSET(R5),- ; CALCULATE AND LOAD THE  
IDBSB\_VECTOR(R5),(R4)[R0] ; VECTOR ADDRESS  
BSBW XI\_DEV\_RESET ; Reset port  
RSB ; Done

```
.SBTTL XI_READ_WRITE, Data transfer FDT
++ XI_READ_WRITE, FDT for READLBLK, READVBLK, READPBLK, WRITELBLK, WRITEVBLK,
      WRITEPBLK
```

Functional description:

- 1) Rejects QUEUE I/O's with odd transfer count
- 2) Rejects QUEUE I/O's for DMA request to UBA Direct Data PATH on odd byte boundary
- 3) Stores request time-out count specified in P3 into IRP
- 4) Stores CTRL bits specified in P4 into IRP
- 5) Checks block mode transfers for memory modify access

Inputs:

```
R3 = Address of IRP
R4 = Address of PCB
R5 = Address of UCB
R6 = Address of CCB
AP = Address of P1
      P1 = Buffer Address
      P2 = Buffer size in bytes
      P3 = Request time-out period (conditional on IOSM_TIMED)
      P4 = Value for CSR (CTRL bits (conditional on IOSM_SETFNCT)
      P5 = 0 for Request A, 1 for Request B (DMA)
```

Outputs:

```
R0 = Error status if odd transfer count
IRPSL_MEDIA = Time-out count for this request
IRPSL_SEGVBN = CTRL bits for PORT CSR
```

--

XI\_READ\_WRITE:

```
10$:  BLBC  P2(AP),20$          ; Branch if transfer count even
      MOVZWL #555 BADPARAM,R0   ; Set error status code
      JMP    G^EXESABORTIO      ; Abort request
20$:  MOVZWL IRPSW_FUNC(R3),R1  ; Fetch I/O Function code
      MOVL   P3(AP),IRPSL MEDIA(R3) ; Set request specific time-out count
      BBS   #IOSV_TIMED,R1,30$    ; Branch if time-out specified
      MOVZWL #XI_DEF_TIMEOUT,-    ; Else set default timeout value
30$:  EXTZV  IRPSL_MEDIA(R3)   ; Get value for CTRL bits
      RSB
```

.SBTTL XI\_SETMODE, Set Mode, Set Char FDT

\*\* XI\_SETMODE, FDT routine to process SET MODE and SET CHARACTERISTICS

#### Functional description:

If **IOSM\_ATTNAST** modifier is set, queue attention AST for device  
Else, finish I/O.

### Inputs:

R3 = I/O packet address

R4 = PCB address

R5 = UCB address

R6 = CCB address

R7 = Function code

AP = QIO Parameter List address

## Outputs:

If **IOSM\_ATTNAST** is specified, queue AST on UCB attention AST list. Else, use exec routine to update device characteristics

### XI\_SETMODE:

MOVZWL IRPSW\_FUNC(R3),R0 ; Get entire function code  
BBC #IOSV\_ATTNAST,R0,20\$ ; Branch if not an ATTN AST

; Attention AST request

|       |                      |                                          |
|-------|----------------------|------------------------------------------|
| PUSHR | #^MCR4,R7>           |                                          |
| MOVAB | UCBSL XI ATTN(R5),R7 | : Address of ATTN AST control block list |
| JSB   | G^COM5SETATTNAST     | : Set up attention AST                   |
| POPR  | #^MCR4,R7>           |                                          |
| BLBC  | RO,30S               | : Branch if error                        |
| BISW  | #UCBSM ATTNAST,-     |                                          |
|       | UCBSW DEVSTS(R5)     | : Flag ATTN AST expected.                |
| BBC   | #UCBSW UNEXPT,-      |                                          |
|       | UCBSW DEVSTS(R5),10S | : Deliver AST if unsolicited interrupt   |
| 10\$: | BSBW XI DEC ATTNAST  |                                          |
|       | JMP G^EXESFINISHIO   | : Thats all for now                      |
| 20\$: | JMP G^EXESSETCHAR    | : Set device characteristics             |
| 30\$: | CLRL R1              | : zero R1                                |
|       | JMP G^EXESABORTIO    | : Abort I/O with R0 as status            |

```
.SBTTL XI_START,      Start I/O routines
```

```
** XI_START - Start a data transfer, set characteristics, enable ATTN AST.
```

#### Functional Description:

This routine has one major function:

- 1) Start an I/O transfer. The CTRL bits in the port CSR are set. If the transfer count is zero, the STATUS bits in the PORT CSR are read and the request completed.

#### Inputs:

```
R3 = Address of the I/O request packet
R5 = Address of the UCB
```

#### Outputs:

```
R0 = final status and number of bytes transferred
R1 = value of CSR STATUS bits
```

--

#### XI\_START:

; Retrieve the address of the device CSR

```
ASSUME IDBSL_CSR EQ 0
MOVL UCBSL_CRB(R5),R4      ; Address of CRB
MOVL ACRBSC_INTD+VÉC$L_IDB(R4),R4
                                ; Address of CSR
```

; Fetch the I/O function code

```
MOVZWL IRPSW_FUNC(R3),R1      ; Get entire function code
MOVW  R1,UCBSW_FUNC(R5)      ; Save FUNC in UCB
EXTZV #IOSV_FCODE,-
      #IOSS_FCODE,R1,R2      ; Extract function field
```

; If subfunction modifier for device reset is set, do one here

```
BBC    S^#IOSV_RESET,R1,40$  ; Branch if not device reset
BSBW  XI_DEV_RESET          ; Reset port
```

; This must be a data transfer function - i.e. READ OR WRITE

; Check to see if this is a zero length transfer.

; If so, only set CSR CTRL bits and return STATUS from CSR

```
40$: TSTW UCBSW_BCNT(R5)      ; Is transfer count zero?
BNEQ 100$                  ; No, continue with data transfer
BBC   S^#IOSV_SETFNCT,R1,60$  ; Set CSR CTRL specified?
DSBINT
SETCTRL
MOVZWL XI_CSR(R4),R1          ; Set CTRL bits in CSR
ENBINT                         ; Save CSR
                                ; Enable Interrupts
```

```

    BRB    70$          ; Skip clearing of R1
60$: CLRL   R1          ; Clear R1
70$: BISW   #XI_CSRSM,IEAB,-
           XI_CSR(R4)      ; Enable device interrupts (A & B)
    MOVZWL #SSS_NORMAL,RO
    REQCOM          ; Set success
                      ; Request done

```

; Do the read or the write

```

100$: MOVZWL UCB$W BCNT(R5),RO      ; Get byte count
       ASHL   #-1,R0,UCBSL_XI_DPR(R5) ; Make byte count into word count
       .SBTTL - word mode tranfer

```

++ WORD MODE -- Process word mode (interrupt per word) transfer

FUNCTIONAL DESCRIPTION:

Data is transferred one word at a time with an interrupt for each word. The request is handled separately for a write (from memory to port and a read (from port to memory). For a write, data is fetched from memory, loaded into the ODR of the port and the system waits for an interrupt. For a read, the system waits for a port interrupt and the INBUF is transferred into memory. If the unsolicited interrupt flag is set, the first word is transferred directly into memory without waiting for an interrupt.

WORD\_MODE:

; Dispatch to separate loops on READ or WRITE

```

10$: CMPB   #IOS_READPBLK,R2      ; Check for read function
     BEQL   WORD_MODE_READ

```

PAGE

++ WORD MODE WRITE -- Write (output) in word mode

FUNCTIONAL DESCRIPTION:

Transfer the requested number of words from user memory to the port OUTBUF one word at a time, wait for interrupt for each word.

WORD\_MODE\_WRITE:

```

10$: BSBW   MOVFRUSER
     DSBINT
     MOVW   R1,XI_OUTBUF(R4)      ; Get two bytes from user buffer
                      ; Lock out interrupts
                      ; Flag interrupt expected
                      ; Move data to port

```

```

BISW #XI_CSR$M_IEAB, -           C 9
      XI_CSR(R4)          ; Set Interrupt Enable (A & B)
SETCTRL                         ; Clear and set CTRL bits

```

: Wait for interrupt, powerfail, or device time-out

```
WFIKPCH XI_TIME_OUTW,IRPSL_MEDIA(R3)
```

: Decrement transfer count, and loop until done

```

IOFORK                         ; Fork to lower IPL
DECW   UCB$L_XI_DPR(R5)        ; All words transferred?
BNEQ   10$                   ; No, loop until finished.

```

: Transfer is done, clear interrupt expected flag and FORK

: All words read or written in WORD MODE. Finish I/O.

RETURN\_STATUS:

```

MOVZWL #SS$ NORMAL,R0          ; Complete success status
MULW3 #2,UCBSL_XI_DPR(R5),R1   ; Calculate actual bytes xfered
SUBW3 R1,UCBSW_BCNT(R5),R1     ; From requested number of bytes
INSV  R1,#16,#T6,R0            ; And place in high word of R0
MOVZWL UCB$W_XI_CSR(R5),R1     ; Return CSR status
BICW  #<XI_CSR$M_CTRL0! -     ; XI_CSR$M_CTRL1>,-
      XI_CSR(R4)              ; Clear CTRL bits
BISW  #XI_CSR$M_IEAB, -        ; Enable device interrupts (A & B)
      XI_CSR(R4)              ; Finish request in exec
REQCOM

```

.PAGE

++ WORD MODE READ -- Read (input) in word mode

FUNCTIONAL DESCRIPTION:

Transfer the requested number of words from the port INBUF into user memory one word at a time, wait for interrupt for each word. If the unexpected (unsolicited) interrupt bit is set, transfer the first (last received) word to memory without waiting for an interrupt.

WORD\_MODE READ:

```
SETIPL UCB$B_DIP$L(R5)        ; Lock out interrupts
```

: If an unexpected (unsolicited) interrupt has occurred, assume it is for this READ request and return value to user buffer without waiting for an interrupt.

```
BBSC  #UCBSV_UNEXPT, -          ; Branch if unexpected interrupt
      UCB$W_DEVSTS(R5),20$
```

10\$: DSBINT

```
BISW  #XI_CSR$M_IEAB, -        ; Set Interrupt Enable (A & B)
      XI_CSR(R4)              ; Clear and set CTRL bits
```

SETCTRL

; Wait for interrupt, powerfail, or device time-out  
WFIKPCH XI\_TIME\_OUTW,IRPSL\_MEDIA(R3)

; Decrement transfer count, and loop until done

20\$: IOFORK ; Fork to lower IPL  
BSBW MOVTOUSER ; Store two bytes into user buffer

; Send interrupt back to sender. Acknowledge we got last word.

DSBINT  
DECW UCB\$L\_XI\_DPR(R5) ; Decrement transfer count  
BNEQ 10\$ ; Loop until all words transferred  
SETCTRL  
ENBINT  
BRW RETURN\_STATUS ; Finish request in common code

PAGE

MOVFRUSER - Routine to fetch two bytes from user buffer.

INPUTS:

R5 = UCB address

OUTPUTS:

R1 = Two bytes of data from users buffer  
Buffer descriptor in UCB is updated.

ENABL LSB

MOVFRUSER:

MOVAL -(SP),R1 ; Address of temporary stack loc  
MOVZBL #2,R2 ; Fetch two bytes  
JSB G^IOC\$MOVFRUSER ; Call exec routine to do the deed  
MOVL (SP)+,R1 ; Retrieve the bytes  
BRB 20\$ ; Update UCB buffer pointers

MOVTOUSER - Routine to store two bytes into users buffer.

INPUTS:

R5 = UCB address  
UCB\$W\_XI\_INBUF(R5) = Location where two bytes are saved

OUTPUTS:

Two bytes are stored in user buffer and buffer descriptor in  
UCB is updated.

MOVTOUSER:

MOVAB UCB\$W\_XI\_INBUF(R5),R1 ; Address of internal buffer

```
MOVZBL #2 R2
JSB G^IOCSMOVTOUSER      ; Call exec
20$: ADDW #2,UCBSW_BOFF(R5) ; Update buffer pointers in UCB
BICW #^{<^X01FF>},UCBSW_BOFF(R5) ; Add two to buffer descriptor
BNEQ 30$                  ; Modulo the page size
ADDL #4,UCBSL_SVAPTE(R5)  ; If NEQ, no page boundary crossed
30$: RSB
.DSABL LSB
```

.SBTTL XI\_TIME\_OUTW, Device time-out routine

\*\* Device TIME-OUT

: Clear port CSR

: Return error status

: Power failure will appear as a device time-out

;--

XI\_TIME\_OUTW: ; Time-out for WORD mode transfer

BSBW XI\_DEV\_RESET ; Reset controller

MOVZWL #555\_TIMEOUT,R0 ; Error status

CLRL R1

CLRW UCBSW\_DEVSTS(R5) ; Clear ATTN AST flags

BICW #<UCBSM\_TIM,- ;

UCBSM\_INT,-

UCBSM\_TIMEOUT,-

UCBSM\_CANCEL,-

UCBSM\_POWER>,-

UCBSW\_STS(R5) ; Clear unit status flags

REQCOM ; Complete I/O in exec

```
.SBTTL XI_INTERRUPT, Interrupt service routine for PORT
++ XI_INTERRUPT, Handles interrupts generated by port
```

#### Functional description:

This routine is entered whenever an interrupt is generated by the port. It checks that an interrupt was expected. If not, it sets the unexpected (unsolicited) interrupt flag. All device registers are read and stored into the UCB. If an interrupt was expected, it calls the driver back at its Wait For Interrupt point. Deliver ATTN AST's if unexpected interrupt.

#### Inputs:

```
00(SP) = Pointer to address of the device IDB
04(SP) = saved R0
08(SP) = saved R1
12(SP) = saved R2
16(SP) = saved R3
20(SP) = saved R4
24(SP) = saved R5
28(SP) = saved PSL
32(SP) = saved PC
```

#### Outputs:

The driver is called at its Wait For Interrupt point if an interrupt was expected. The current value of the port CSR's are stored in the UCB.

```
-- XI_INTERRUPT: ; Interrupt service for PORT
    MOVL  a(SP)+,R4 ; Address of IDB and pop SP
    MOVQ  (R4),R4 ; CSR and UCB address from IDB
; Read INBUF and CSR
    MOVW  XI_INBUF(R4), -UCBSW_XI_INBUF(R5) ; Read input data
    MOVW  XI_CSR(R4), -UCBSW_XI_CSR(R5) ; Read CSR
; Check to see if device transfer request active or not
; If so, call driver back at Wait for Interrupt point and
; Clear unexpected interrupt flag.
    BBCC  #UCBSV_INT, -UCBSW_STS(R5),10$ ; If clear, no interrupt expected
; Interrupt expected, clear unexpected interrupt flag and call driver
; back.
    BICW  #UCBSM_UNEXPT, -
```

```
MOVL  UCBSW_DEVSTS(R5)      ; Clear unexpected interrupt flag
JSB   UCBSL_FR3(R5) R3      ; Restore drivers R3
BRB   20$                  ; Call driver back after WFIKPCH
                                ; Exit

; Deliver ATTN AST's if no interrupt expected and set unexpected
; interrupt flag.

10$: BISW  #UCBSM_UNEXPT,-    ; Set unexpected interrupt flag
      UCBSW_DEVSTS(R5)
BSBW  XI_DEC_ATTNAST       ; Deliver ATTN AST's
BISW  #XI_CSRSM_IEAB,-    ; Enable device interrupts (A & B)
      XI_CSR(R4)

; Restore registers and return from interrupt

20$: POPR  #^M<R0,R1,R2,R3,R4,R5> ; Restore registers
REI                           ; Return from interrupt
```

.SBTTL XI\_CANCEL, Cancel I/O routine

++ XI\_CANCEL, Cancels an I/O operation in progress

Functional description:

Flushes Attention AST queue for the user.  
If transfer in progress, do a device reset to port  
and finish the request.  
Clear interrupt expected flag.

Inputs:

R2 = negated value of channel index  
R3 = address of current IRP  
R4 = address of the PCB requesting the cancel  
R5 = address of the device's UCB

Outputs:

;--

XI\_CANCEL: ; Cancel I/O

BBCC #UCBSV\_ATTNAST,-  
UCBSW\_DEVSTS(R5),20\$ ; ATTN AST enabled?

; Finish all ATTN AST's for this process.

PUSHR #^M<R2,R6,R7>  
MOVL R2,R6 ; Set up channel number  
MOVAB UCBSL XI\_ATTN(R5),R7 ; Address of listhead  
JSB G^COMSFL0SHATTNS ; Flush ATTN AST's for process  
POPR #^M<R2,R6,R7>  
BICW #UCBSM\_UNEXPT,-  
UCBSW\_DEVSTS(R5) ; Clear unexpected interrupt flag

; Check to see if a data transfer request is in progress  
; for this process on this channel

20\$:

SETIPL UCBSB\_DIPL(R5) ; Lock out device interrupts  
JSB G^IOCSCANCELIO ; Check if transfer going  
BBC #UCBSV\_CANCEL,-  
UCBSW\_STS(R5),30\$ ; Branch if not for this guy  
MOVZWL #SSS\_CANCEL,R0 ; Status is request canceled  
CLRL R1  
CLRW UCBSW\_DEVSTS(R5) ; Clear unexpected interrupt flag  
BICW #<UCBSM\_TIM  
UCBSM\_BSY  
UCBSM\_CANCEL  
UCBSM\_INT  
UCBSM\_TIMEOUT>,-  
UCBSW\_STS(R5) ; Clear unit status flags  
REQCOM ; Jump to exec to finish I/O

30\$:

```
SETIPL UCB$B_FIPL(R5)      ; Lower to FORK IPL
RSB          ; Return
```

.SBTTL XI\_DEL\_ATTNAST, Deliver ATTN AST's  
 \*\* XI\_DEL\_ATTNAST, Deliver all outstanding ATTN AST's

## Functional description:

This routine is used by the port driver to deliver all of the outstanding attention AST's. It is copied from COMSDELATTNAST in the exec. In addition, it places the saved value of the port CSR and Input Data Buffer Register in the AST parameter.

## Inputs:

R5 = UCB of unit

## Outputs:

R0,R1,R2 Destroyed  
 R3,R4,R5 Preserved

## XI\_DEL\_ATTNAST:

|       |                            |                                      |
|-------|----------------------------|--------------------------------------|
| BBCC  | #UCBSV_ATTNAST, -          |                                      |
|       | UCBSW_DEVSTS(R5),30\$      | : Any ATTN AST's expected?           |
| 10\$: | PUSHR #^M<R3,R4,R5>        | : Save R3,R4,R5                      |
|       | MOVL 8(SP),R1              | : Get address of UCB                 |
|       | MOVAB UCBSL_XI_ATTN(R1),R2 | : Address of ATTN AST listhead       |
|       | MOVL (R2),R5               | : Address of next entry on list      |
|       | BEQL 20\$                  | : No next entry, end of loop         |
|       | BICW #UCBSM_UNEXPT, -      |                                      |
|       | UCBSW_DEVSTS(R1)           | : Clear unexpected interrupt flag    |
|       | MOVL (R5),TR2              | : Close list                         |
|       | MOVW UCBSW_XI_INBUF(R1), - |                                      |
|       | ACBSL_KAST+6(R5)           | : Store INBUF in AST parameter       |
|       | MOVW UCBSW_XI_CSR(R1), -   |                                      |
|       | ACBSL_KAST+4(R5)           | : Store CSR in AST parameter         |
|       | PUSHAB B^10\$              | : Set return address for FORK        |
|       |                            | : so that it loops through all AST's |
|       | FORK                       | : FORK for this AST                  |

## : AST fork procedure

|        |                                 |                             |
|--------|---------------------------------|-----------------------------|
| MOVQ   | ACBSL_KAST(R5),ACBSL_AST(R5)    |                             |
|        |                                 | : Re-arrange entries        |
| MOVB   | ACBSL_KAST+8(R5),ACBSB_RMOD(R5) |                             |
| MOVL   | ACBSL_KAST+12(R5),ACBSL_PID(R5) |                             |
| CLRL   | ACBSL_KAST(R5)                  |                             |
| MOVZBL | #PRIS_I0COM,R2                  | : Set up priority increment |
| JMP    | G^SCH\$QAST                     | : Queue the AST             |
| 20\$:  | POPR #^M<R3,R4,R5>              | : Restore registers         |
| 30\$:  | RSB                             | : Return                    |

.SBTTL XI\_DEV\_RESET. Device reset routine

\*\* XI\_DEV\_RESET - Device reset routine

This routine raises IPL to device IPL, performs a device reset to the required controller, and re-enables device interrupts.

Inputs:

R4 - Address of Control and Status Register  
R5 - Address of UCB

Outputs:

Controller is reset, controller interrupts are enabled

-- XI\_DEV\_RESET:

DSBINT ; Raise IPL to lock all interrupts

BISW #XI\_CSRSM\_RESET,-  
XI\_CSR(R4) ; Reset device

TIMEWAIT -  
TIME = #500,-  
BITVAL = #XI\_CSRSM\_RESET,-  
SOURCE = XI\_CSR(R4),-  
CONTEXT = W,-  
SENSE = .FALSE.

BISW #XI\_CSRSM\_IEAB,-  
XI\_CSR(R4) ; Enable device interrupts (A & B)

ENBINT  
RSB ; Restore IPL

XI-END: .END ; End of driver label

0107 AH-BT13A-SE  
VAX/VMS V4.0

DIGITAL EQUIPMENT CORPORATION  
CONFIDENTIAL AND PROPRIETARY

XDRIVER  
MAR

XDRIVER  
MAR

XDRIVER  
LIS

PAMAC  
MAR

CONINTERR  
LIS